home *** CD-ROM | disk | FTP | other *** search
/ MacWorld 1999 March - Disc 1 / Macworld (1999-03) (Disk 1).dmg / Shareware World / Utilities / Text Processing / Alpha / Tcl / Modes / latex Mode / latexComm.tcl < prev    next >
Encoding:
Text File  |  1998-12-19  |  15.6 KB  |  549 lines  |  [TEXT/ALFA]

  1. #############################################################################
  2. #############################################################################
  3. #
  4. # latexComm.tcl (called from latex.tcl)
  5. #
  6. #############################################################################
  7. #
  8. # Old Author:  Tom Scavo <trscavo@syr.edu>
  9. # New Author:  Vince Darley <darley@fas.harvard.edu>
  10. #############################################################################
  11. #############################################################################
  12.  
  13. proc latexComm.tcl {} {}
  14. #--------------------------------------------------------------------------
  15. # TeX applications
  16. #--------------------------------------------------------------------------
  17.  
  18. # In the following scripts, $quotedSig and $filename are the application 
  19. # signature and the name of the file to be typeset, respectively.  (See
  20. # proc 'evalTeXScript' below.)
  21. array set bibtexAppScripts {
  22.   DirectTeXPro {
  23.     {dosc -c $quotedSig -s  "ProjectMgr -t \$dt_TeXFormat    \"[win::Current]\"; MenuCommand 1 5"}
  24. } CMacTeX {
  25.     {sendOpenEvent noReply $quotedSig $filename}
  26. } BibTeX {
  27.     {sendOpenEvent noReply $quotedSig $filename}
  28. }
  29. }
  30. array set bibtexAppSignatures {
  31.   CMacTeX CMTu BibTeX Vbib
  32. }
  33. array set dvipsAppScripts {
  34.   DirectTeXPro {
  35.     {dosc -c $quotedSig -s  "ProjectMgr -t \$dt_TeXFormat    \"[win::Current]\";  directory \$dt_TeXProjectDir > CurrDirectory; dvips \$dt_TeXProjectName"}
  36. } CMacTeX {
  37.     {dosc -c $quotedSig -k 'aevt' -e 'odoc' -t 600 -f $filename}
  38. } OzTeX {
  39.     {sendOpenEvent noReply $quotedSig $filename}
  40. }
  41. }
  42. array set dvipsAppSignatures {
  43.   DirectTeXPro TeX+ CMacTeX CMT1 OzTeX OzDP
  44. }
  45. array set makeindexAppScripts {
  46.   DirectTeXPro {
  47.     {dosc -c $quotedSig -s  "ProjectMgr -t \$dt_TeXFormat    \"[win::Current]\"; MenuCommand 1 6"}
  48. } CMacTeX {
  49.     {sendOpenEvent noReply $quotedSig $filename}
  50. } MakeIndex {
  51.     {sendOpenEvent noReply $quotedSig $filename}
  52. }
  53. }
  54. array set makeindexAppSignatures {
  55.   DirectTeXPro TeX+ CMacTeX CMTt MakeIndex RZMI
  56. }
  57. array set printDVIAppScripts {
  58.   Textures {
  59.     {AEBuild $quotedSig aevt pdoc "----" [makeAlis $filename]}
  60. } DirectTeXPro {
  61.     {dosc -c $quotedSig -s  "ProjectMgr -t \$dt_TeXFormat    \"[win::Current]\"; MenuCommand 1 9"}
  62. } CMacTeX {
  63.     {dosc -c $quotedSig -k 'aevt' -e 'pdoc' -r -f $filename}
  64. } OzTeX {
  65.     {dosc -c $quotedSig -k 'aevt' -e 'pdoc' -r -f $filename}
  66. }
  67. }
  68. array set printDVIAppSignatures {
  69.   Textures *TEX DirectTeXPro TeX+ CMacTeX CMT8 OzTeX OTEX
  70. }
  71. array set printPSAppScripts {
  72.   DropPS {
  73.     {dosc -c $quotedSig -k 'aevt' -e 'odoc' -r -f $filename}
  74. } DirectTeXPro {
  75.     {dosc -c $quotedSig -s  "ProjectMgr -t \$dt_TeXFormat    \"[win::Current]\";  directory \$dt_TeXProjectDir > CurrDirectory; download \$dt_TeXProjectName.ps"}
  76. } CMacTeX {
  77.     {dosc -c $quotedSig -k 'aevt' -e 'odoc' -r -f $filename}
  78. }
  79. }
  80. array set printPSAppSignatures {
  81.   DropPS D•PS DirectTeXPro TeX+ CMacTeX PSP*
  82. }
  83. array set texAppScripts {
  84.   DirectTeX {
  85.     {set script "Begin; ChangeTeXProject '$filename' -check -confirm || Exit 0;  Execute \"{dt_TeXProject}\"; TeXMenu -tex -formats;  End ∑ Dev:Null; RunSession 1 ∑ Dev:Null"} 
  86.     {dosc -r -c $quotedSig -s $script} 
  87. } Textures {
  88.     {global Texturesconnections} 
  89.     {set TeXjob ""} 
  90.     {
  91.         if {[info exists Texturesconnections]} {
  92.             foreach entry $Texturesconnections {
  93.                 if {[car $entry] == $filename} {
  94.                     set TeXjob [cadr $entry]
  95.                     break
  96.                 }
  97.             }
  98.         }
  99.     }   
  100.     {
  101.         if { $TeXjob == "" } {
  102.             set TeXjob [AEBuild -r $quotedSig BSRs Begi]
  103.             set TeXjob [string trim [string range $TeXjob 15 end] {\{\}\"}]
  104.             lappend Texturesconnections [list $filename $TeXjob]
  105.         }
  106.     }   
  107.     {AEBuild -t 1200 $quotedSig BSRs Typs "----" [makeAlis $filename] fmat {"LaTeX"} Jobi $TeXjob} 
  108. } DirectTeXPro {
  109.     {dosc -c $quotedSig -s  "if \$dt_TeXFormat = \'\' \"set dt_TeXFormat -x \$dt_DefaultFormat\""} 
  110.     {dosc -c $quotedSig -s  "ProjectMgr -t \$dt_TeXFormat \"$filename\"; MenuCommand 1 4"} 
  111. } CMacTeX {
  112.     {dosc -c $quotedSig -k 'aevt' -e 'odoc' -r -f $filename}
  113. } OzTeX {
  114.     {sendOpenEvent noReply $quotedSig $filename}
  115. }
  116. array set texAppSignatures {
  117.   DirectTeX MPS* Textures *TEX DirectTeXPro TeX+ 
  118.   CMacTeX {*XeT BTxT pXeT} OzTeX OTEX
  119. }
  120. array set viewDVIAppScripts {
  121.   Textures {
  122.     {AEBuild $quotedSig aevt odoc "----" [makeAlis $filename]}
  123. } DirectTeXPro {
  124.     {dosc -c $quotedSig -s "ProjectMgr -t \$dt_TeXFormat    \"[win::Current]\"; MenuCommand 1 8"}
  125. } CMacTeX {
  126.     {dosc -c $quotedSig -k 'aevt' -e 'odoc' -r -f $filename}
  127. } OzTeX {
  128.     {sendOpenEvent noReply $quotedSig $filename}
  129. }
  130. }
  131. array set viewDVIAppSignatures {
  132.   Textures *TEX DirectTeXPro TeX+ CMacTeX {PIVD CMT8} OzTeX OTEX
  133. }
  134. array set viewPSAppScripts {
  135.   CMacTeX {
  136.     {dosc -c $quotedSig -k 'aevt' -e 'odoc' -r -f $filename}
  137. } MacGS {
  138.     {dosc -c $quotedSig -k 'aevt' -e 'odoc' -r -f $filename}
  139. }
  140. }
  141. array set viewPSAppSignatures {
  142.   CMacTeX {CMT5 CMTJ TMCJ} MacGS gsVR
  143. }
  144.  
  145. #--------------------------------------------------------------------------
  146. # Typeset submenu commands
  147. #--------------------------------------------------------------------------
  148.  
  149. proc typeset {{bg 0}} {
  150.     # Is there a window open?
  151.     set currentWin [win::Current]
  152.     if { $currentWin == "" } {
  153.         typesetFile "" $bg
  154.         return
  155.     }
  156.     # Strip off trailing garbage (if any):
  157.     regexp {(.*) <[0-9]+>} $currentWin dummy currentWin
  158.     # Is the current window part of TeX fileset?
  159.     set fset [isWindowInFileset $currentWin "tex"]
  160.     if { $fset != "" } {
  161.         if [dirtyFileset $fset] {
  162.             switch [askyesno -c "Save current TeX fileset?"] {
  163.                 "yes" {saveEntireFileset $fset}
  164.                 "no" {
  165.                      # do nothing
  166.                 }
  167.                 "cancel" {return}
  168.             }
  169.         }
  170.         typesetFile [texFilesetBaseName $fset] $bg
  171.         return
  172.     }
  173.     # Is the window untitled or dirty?
  174.     global PREFS
  175.     set currentDoc [file tail $currentWin]
  176.     if { [set num [winUntitled]] } {
  177.         switch [askyesno -c "Save \"$currentDoc\"?"] {
  178.             "yes" {
  179.                 if {[catch {set currentWin [saveAs "Untitled$num\.tex"]}]} {return}
  180.             }
  181.             "no" {
  182.                 set tmpFilename "Untitled$num\.tex"
  183.                 set text [getText [minPos] [maxPos]]
  184.                 if {![file exists [file join $PREFS tmp]]} { mkdir [file join $PREFS tmp] }
  185.                 set newDoc [file join $PREFS tmp $tmpFilename]
  186.                 file::writeAll $newDoc $text 1
  187.                 set currentWin $newDoc
  188.             }
  189.             "cancel" {return}
  190.         }
  191.     } elseif { [winDirty] } {
  192.         switch [askyesno -c "Save \"$currentDoc\"?"] {
  193.             "yes" {save}
  194.             "no" {
  195.                 set text [getText [minPos] [maxPos]]
  196.                 if {![file exists [file join $PREFS tmp]]} { file mkdir [file join $PREFS tmp] }
  197.                 set newDoc [file join $PREFS tmp temp-$currentDoc]
  198.                 file::writeAll $newDoc $text 1
  199.                 set currentWin $newDoc
  200.             }
  201.             "cancel" {return}
  202.         }
  203.     }
  204.     # Is the current window TeX-able?
  205.     set ext [file extension $currentWin]
  206.     if { [lsearch -exact {.tex .ltx .dtx .ins} $ext] < 0 } {
  207.         typesetFile "" $bg
  208.         return
  209.     }
  210.     typesetFile $currentWin $bg
  211. }
  212.  
  213. proc typesetSelection {} {
  214.     global PREFS
  215.     if { [isSelection] } { 
  216.         watchCursor
  217.         message "processing selection…"
  218.         set currentWin [win::Current]
  219.         # Is the current window part of TeX fileset?
  220.         set fset [isWindowInFileset $currentWin "tex"]
  221.         if { $fset == "" } {
  222.             set pos1 [getPos]
  223.             if { [isInDocument] } {
  224.                 set pos2 [selEnd]
  225.                 if { [isInDocument] } {
  226.                     # fall through
  227.                 } else {
  228.                     beep
  229.                     set msg "Selection not in document environment.  Continue?"
  230.                     if { [askyesno $msg] == "no" } { return }
  231.                 }
  232.             } else {
  233.                 beep
  234.                 set msg "Selection not in document environment.  Continue?"
  235.                 if { [askyesno $msg] == "no" } { return }
  236.                 set pos2 [selEnd]
  237.             }
  238.             set body "\r[getText $pos1 $pos2]\r"
  239.             set searchText [getText [minPos] [maxPos]]
  240.         } else {
  241.             set body "\r[getSelect]\r"
  242.             # Will not handle a base file that is open and dirty:
  243.             set searchText [buildFilecontents [texFilesetBaseName $fset]]
  244.         }
  245.         message "building temporary document…"
  246.         set pattern {(\\documentclass.*)\\begin\{document\}}
  247.         if { ![regexp $pattern $searchText dummy preamble] } {
  248.             set preamble "\\documentclass\{article\}\r"
  249.         }
  250.         set rootFile [file rootname $currentWin]
  251.         set tempFile "temp-[file tail $rootFile]"
  252.         set auxFile $rootFile.aux
  253.         if { [file exists $auxFile] } {
  254.             set latexDoc [buildFilecontents $auxFile $tempFile.aux]
  255.         } else {
  256.             set latexDoc {}
  257.         }
  258.         append latexDoc $preamble [buildEnvironment "document" "" $body "\r"]
  259.         set currentDir [file dirname $currentWin]
  260.         set latexDoc [texResolveAll $latexDoc $currentDir]
  261.         if {![file exists [file join $PREFS tmp]]} { file mkdir [file join $PREFS tmp] }
  262.         set newFile [file join $PREFS tmp $tempFile.tex]
  263.         file::writeAll $newFile $latexDoc 1
  264.         typesetFile $newFile
  265.     } else {
  266.         beep
  267.         message "no selection"
  268.     }
  269. }
  270.  
  271. proc typesetClipboard {} {
  272.     global PREFS
  273.     set body "\r[getScrap]\r"
  274.     set pat1 {\\begin\{document\}.*\\end\{document\}}
  275.     set pat2 {\\documentclass}
  276.     set preamble "\\documentclass\{article\}\r"
  277.     # Check to see if there's a document environment:
  278.     if {![regexp $pat1 $body]} {
  279.         append text $preamble [buildEnvironment "document" "" $body "\r"]
  280.     } else {
  281.         # Check to see if there's a \documentclass command:
  282.         if {![regexp $pat2 $body]} {
  283.             append text $preamble $body
  284.         } else {
  285.             set text $body
  286.         }
  287.     }
  288.     if {![file exists [file join $PREFS tmp]]} { file mkdir [file join $PREFS tmp] }
  289.     set newFile [file join $PREFS tmp temp-noname\.tex]
  290.     file::writeAll $newFile $text 1
  291.     set currentWin $newFile
  292.     typesetFile $currentWin
  293. }
  294.  
  295. # Typeset $filename, but perform no error-checking
  296. #
  297. proc typesetFile {filename {bg 0}} {
  298.     if { $filename == "" } {
  299.     set filename [getfile "Choose a file to typeset:"]
  300.     }
  301.     evalTeXScript tex {TeX app} $filename $bg
  302. }
  303.  
  304. # Apply $op to a file with extension $ext.  (See latexMenu.tcl for
  305. # many examples.)  If 'forcecurrent == 1', use the current window 
  306. # even if it belongs to a TeX fileset.
  307. #
  308. proc doTypesetCommand {op ext {forcecurrent 0}} {
  309.     if { [set filename [findAuxiliaryFile $ext $forcecurrent]] != "" } {
  310.     if { $op == "open" } { 
  311.         edit -r -m -w $filename 
  312.     } else {
  313.         $op${ext}File $filename
  314.     }
  315.     } else {
  316.     beep
  317.     alertnote "No $ext file found!"
  318.     }
  319. }
  320.  
  321. proc viewDVIFile {filename} {
  322.     evalTeXScript viewDVI {DVI viewer} $filename
  323. }
  324.  
  325. proc printDVIFile {filename} {
  326.     evalTeXScript printDVI {DVI print driver} $filename
  327. }
  328.  
  329. proc dvipsDVIFile {filename} {
  330.     evalTeXScript dvips {DVI-to-PS filter} $filename
  331. }
  332.  
  333. proc viewPSFile {filename} {
  334.     if {[catch {evalTeXScript viewPS {PS viewer} $filename}]} {
  335.     message "View aborted."
  336.     }
  337. }
  338.  
  339. proc printPSFile {filename} {
  340.     evalTeXScript printPS {PS print driver} $filename
  341. }
  342.  
  343. proc bibtexAUXFile {filename} {
  344.     evalTeXScript bibtex {BibTeX app} $filename
  345. }
  346.  
  347. proc makeindexIDXFile {filename} {
  348.     evalTeXScript makeindex {MakeIndex app} $filename
  349. }
  350.  
  351. proc makeindexGLOFile {filename} {
  352.     evalTeXScript makeindex {MakeIndex app} $filename
  353. }
  354.  
  355. proc evalTeXScript {op prompt filename {runAppInBackground 0}} {
  356.     global showTeXLog
  357.     if {[info exists showTeXLog]} {
  358.     app::runScript $op $prompt $filename $runAppInBackground $showTeXLog
  359.     } else {
  360.     app::runScript $op $prompt $filename $runAppInBackground 0
  361.     }
  362.     # this code needs a mechanism for waiting for the TeX run to complete
  363.     if {0 && !$runAppInBackground && ($showTeXLog == 2)} {
  364.     set fname [file tail $filename]
  365.     set curr [file tail [win::Current]]
  366.     if {[file extension $curr] == ".log" && ([file root $fname] == [file root $curr])} {
  367.         return
  368.     } else {
  369.         set fname [file root $filename]
  370.         file::openQuietly ${fname}.log
  371.     }
  372.     }
  373. }
  374.  
  375. # Two bugs in 'getfile' (see "alpha.bugs58"):
  376. proc openAnyTeXFile {} {
  377.     set    filename [getfile "" [win::Current]]
  378.     if { ![string length $filename] } return
  379.     edit -r -m -w $filename
  380. }
  381.  
  382. proc removeAuxiliaryFiles {} {
  383.     set word ""
  384.     set removeSilently 0
  385.     set currentWin [win::Current]
  386.     if { $currentWin == "" } { return }
  387.     set currentDir [file dirname $currentWin]
  388.     set extensions {.ps .dvi .log .aux .bbl .idx .ind .glo .gls \
  389.                     .toc .lof .lot .blg .ilg}
  390.     foreach ext $extensions {
  391.         message "Checking for *$ext files…"
  392.         set files [glob -nocomplain [file join $currentDir *$ext]]
  393.         foreach file $files {
  394.             set word " more"
  395.             if {$removeSilently} {
  396.                 if {[catch {rm [file join $currentDir *$ext]}]} {
  397.                     alertnote "not all \"*$ext\" files deleted"
  398.                 }
  399.                 break
  400.             } else {
  401.                 message ""
  402.                 set filename [file tail $file]
  403.                 switch [buttonAlert "Remove \"$filename\"?" "yes" "no" {rm ext} {rm all} "cancel"] {
  404.                     "yes" {
  405.                         message "Removing $filename…"
  406.                         if {[catch {removeFile [file join $currentDir $filename]}]} {
  407.                             alertnote "\"$filename\" not deleted"
  408.                         } else {
  409.                             message $filename
  410.                         }
  411.                     }
  412.                     "no" {}
  413.                     "rm ext" {
  414.                         if {[catch {rm [file join $currentDir *$ext]}]} {
  415.                             alertnote "not all \"*$ext\" files deleted"
  416.                         }
  417.                         break
  418.                     }
  419.                     "rm all" {
  420.                         if {[catch {rm [file join $currentDir *$ext]}]} {
  421.                             alertnote "not all \"*$ext\" files deleted"
  422.                         }
  423.                         set removeSilently 1
  424.                         break
  425.                     }
  426.                     "cancel" {return}
  427.                 }
  428.             }
  429.         }
  430.     }
  431.     message "No$word files found"
  432. }
  433.  
  434. #--------------------------------------------------------------------------
  435. # Utility procs:
  436. #--------------------------------------------------------------------------
  437.  
  438. # Find a LaTeX auxiliary file with extension $ext.  If 'forcecurrent' 
  439. # is true, search the current directory without checking for TeX filesets.
  440. #
  441. proc findAuxiliaryFile {ext {forcecurrent 0}} {
  442.     
  443.     set currentWin [win::Current]
  444.     if { $currentWin == "" } { return "" }
  445.     set currentDoc [file tail $currentWin]
  446.  
  447.     if $forcecurrent {
  448.         # pretend there are no TeX filesets:
  449.         set fset ""
  450.     } else {
  451.         set fset [isWindowInFileset $currentWin "tex"]
  452.     }
  453.     
  454.     if { $fset != "" } {
  455.         
  456.         set currentWin [texFilesetBaseName $fset]
  457.         set currentDoc [file tail $currentWin]
  458.         set currentDir [file dirname $currentWin]
  459.         set docBasename [file rootname $currentDoc]
  460.         set lowerExt [string tolower $ext]
  461.         
  462.     } else {
  463.         
  464.         # we do all this if it's not a project:
  465.         set currentDir [file dirname $currentWin]        
  466.         set docBasename [file rootname $currentDoc]
  467.         set lowerExt [string tolower $ext]
  468.         
  469.         # Is the window untitled or dirty?
  470.         global PREFS
  471.         if { [set num [winUntitled]] } {
  472.             set filename [file join $PREFS tmp Untitled$num\.$lowerExt]
  473.             if { [file exists $filename] } {
  474.                 return $filename
  475.             } else {
  476.                 return ""
  477.             }
  478.         } elseif { [winDirty] } {
  479.             switch [askyesno "Window dirty---continue anyway?"] {
  480.                 "yes" {
  481.                     set filename [file join $PREFS tmp temp-$currentDoc\.$lowerExt]
  482.                     if { [file exists $filename] } {
  483.                         return $filename
  484.                     } else {
  485.                         # fall through
  486.                     }
  487.                 }
  488.                 "no" {return ""}
  489.             }
  490.         }
  491.     }
  492.     
  493.     # Check the current directory:
  494.     set filename [file join $currentDir $docBasename\.$lowerExt]
  495.     if { [file exists $filename] } {
  496.         return $filename
  497.     } else {
  498.         return ""
  499.     }
  500. }
  501.  
  502. # If the current window is untitled, return its number (i.e., either
  503. # the number 1 or the number  n  in "Untitled <n>"); otherwise, return 0.
  504. proc winUntitled {} {
  505.     set currentWin [win::Current]
  506.     if { $currentWin == "" } { return 0 }
  507.     set currentDoc [file tail $currentWin]
  508.     if { [string match $currentWin $currentDoc] } {
  509.         if { [regexp {<(.*)>} $currentDoc dummy num] } {
  510.             return $num
  511.         } else {
  512.             return 1
  513.         }
  514.     } else {
  515.         return 0
  516.     }
  517. }
  518.  
  519. ## 
  520.  # -------------------------------------------------------------------------
  521.  # 
  522.  # "texApp" --
  523.  # 
  524.  #  Switch to bibtex, latex or makeindex
  525.  # -------------------------------------------------------------------------
  526.  ##
  527. proc texApp {name} {
  528.     set type [string tolower $name]
  529.     global ${type}Sig ${type}AppSignatures
  530.     set supportedApps [array names ${type}AppSignatures]
  531.     foreach app $supportedApps { eval lappend sigs [set ${type}AppSignatures($app)] }
  532.     set longPrompt "Please locate your ${name} app."
  533.     if { [catch {app::launchAnyOfThese $sigs ${type}Sig $longPrompt} appname] } {
  534.     error "bug in 'app::launchAnyOfThese'"
  535.     }
  536.     set quotedSig "'[string trim [set ${type}Sig] {'}]'"
  537.     switchTo $quotedSig
  538. }
  539.  
  540.  
  541.  
  542.  
  543.  
  544.  
  545.  
  546.  
  547.  
  548.